/*******************************************************************************
 * Copyright IBM Corp. and others 2018
 *
 * This program and the accompanying materials are made available under
 * the terms of the Eclipse Public License 2.0 which accompanies this
 * distribution and is available at https://www.eclipse.org/legal/epl-2.0/
 * or the Apache License, Version 2.0 which accompanies this distribution
 * and is available at https://www.apache.org/licenses/LICENSE-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License, v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception [1] and GNU General Public
 * License, version 2 with the OpenJDK Assembly Exception [2].
 *
 * [1] https://www.gnu.org/software/classpath/license.html
 * [2] https://openjdk.org/legal/assembly-exception.html
 *
 * SPDX-License-Identifier: EPL-2.0 OR Apache-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0 OR GPL-2.0-only WITH OpenJDK-assembly-exception-1.0
 *******************************************************************************/

#ifndef OMR_ARM64_SYSTEMLINKAGE_INCL
#define OMR_ARM64_SYSTEMLINKAGE_INCL

#include "codegen/Linkage.hpp"

#include <stdint.h>
#include "codegen/Register.hpp"

namespace TR {
class AutomaticSymbol;
class CodeGenerator;
class Instruction;
class Node;
class ParameterSymbol;
class RegisterDependencyConditions;
class ResolvedMethodSymbol;
} // namespace TR
template<class T> class List;

namespace TR {

class ARM64SystemLinkage : public TR::Linkage {
protected:
    TR::ARM64LinkageProperties _properties;

public:
    /**
     * @brief Constructor
     * @param[in] cg : CodeGenerator
     */
    ARM64SystemLinkage(TR::CodeGenerator *cg);

    /**
     * @brief Gets linkage properties
     * @return Linkage properties
     */
    virtual const TR::ARM64LinkageProperties &getProperties();

    /**
     * @brief Gets the RightToLeft flag
     * @return RightToLeft flag
     */
    virtual uint32_t getRightToLeft();
    /**
     * @brief Maps symbols to locations on stack
     * @param[in] method : method for which symbols are mapped on stack
     */
    virtual void mapStack(TR::ResolvedMethodSymbol *method);
    /**
     * @brief Maps an automatic symbol to an index on stack
     * @param[in] p : automatic symbol
     * @param[in/out] stackIndex : index on stack
     */
    virtual void mapSingleAutomatic(TR::AutomaticSymbol *p, uint32_t &stackIndex);
    /**
     * @brief Initializes ARM64 RealRegister linkage
     */
    virtual void initARM64RealRegisterLinkage();

    /**
     * @brief Creates method prologue
     * @param[in] cursor : instruction cursor
     */
    virtual void createPrologue(TR::Instruction *cursor);
    /**
     * @brief Creates method prologue
     * @param[in] cursor : instruction cursor
     */
    virtual void createPrologue(TR::Instruction *cursor, List<TR::ParameterSymbol> &parm);

    /**
     * @brief Creates method epilogue
     * @param[in] cursor : instruction cursor
     */
    virtual void createEpilogue(TR::Instruction *cursor);

    /**
     * @brief Builds method arguments
     * @param[in] node : caller node
     * @param[in] dependencies : register dependency conditions
     */
    virtual int32_t buildArgs(TR::Node *callNode, TR::RegisterDependencyConditions *dependencies);

    /**
     * @brief Builds direct dispatch to method
     * @param[in] node : caller node
     */
    virtual TR::Register *buildDirectDispatch(TR::Node *callNode);

    /**
     * @brief Builds indirect dispatch to method
     * @param[in] node : caller node
     */
    virtual TR::Register *buildIndirectDispatch(TR::Node *callNode);

    /**
     * @brief Provides the entry point in a method to use when that method is invoked
     *        from a method compiled with the same linkage.
     *
     * @details
     *    When asked on the method currently being compiled, this API will return 0 if
     *    asked before code memory has been allocated.
     *
     *    The compiled method entry point may be the same as the interpreter entry point.
     *
     * @return The entry point for compiled methods to use; 0 if the entry point is unknown
     */
    virtual intptr_t entryPointFromCompiledMethod();

    /**
     * @brief Provides the entry point in a method to use when that method is invoked
     *        from an interpreter using the same linkage.
     *
     * @details
     *    When asked on the method currently being compiled, this API will return 0 if
     *    asked before code memory has been allocated.
     *
     *    The compiled method entry point may be the same as the interpreter entry point.
     *
     * @return The entry point for interpreted methods to use; 0 if the entry point is unknown
     */
    virtual intptr_t entryPointFromInterpretedMethod();

    /**
     * @brief Sets parameter linkage register index
     * @param[in] method : method symbol
     */
    virtual void setParameterLinkageRegisterIndex(TR::ResolvedMethodSymbol *method);

private:
    // Tactical GRA
    static uint32_t _globalRegisterNumberToRealRegisterMap[];
};

} // namespace TR

#endif
